package com.qd.panda.service.card.third;

import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.cdqidi.constant.ConstantDto;
import com.cdqidi.db.BaseService;
import com.cdqidi.dto.CacheKey;
import com.cdqidi.exception.ApiException;
import com.cdqidi.util.RedisTemplateUtil;
import com.qd.common.panda.domain.entity.card.third.ThirdPartyDTO;
import com.qd.common.sys.enums.BaseTableEnum;
import com.qd.panda.mapper.card.ThirdPartyMapper;
import com.qd.panda.model.ThirdParty;
import com.qd.system.service.dict.MbUtil;
import com.qd.system.service.login.LoginUserService;
import com.qd.system.service.org.OrgService;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.StringUtils;

import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Optional;

import static java.util.stream.Collectors.toList;

/**
 * 第三方券商
 *
 * @author sjk
 */
@RequiredArgsConstructor
@Service
@Slf4j
public class ThirdPartyService extends BaseService<ThirdPartyMapper, ThirdParty> {

    private static final String PREFIX_ID = "panda_third_party_code_{0}";

    private final RedisTemplateUtil<ThirdParty> redisTemplateUtil;
    private final OrgService orgService;
    private final LoginUserService loginUserService;

    /**
     * 保存
     *
     * @param dto 页面对象
     * @return 页面对象
     */
    @Transactional(rollbackFor = Exception.class)
    public ThirdPartyDTO save(ThirdPartyDTO dto) {
        validator(dto);
        final ThirdParty model = dtoToModel(dto);
        try {
            if (super.save(model)) {
                return modelToDto(model);
            }
            throw new ApiException("添加失败");
        } finally {
            clearCache(model);
            model.freeData();
        }
    }

    /**
     * 修改
     *
     * @param dto 页面对象
     * @return 页面对象
     */
    @Transactional(rollbackFor = Exception.class)
    public ThirdPartyDTO update(ThirdPartyDTO dto) {
        validator(dto);
        final ThirdParty historyModel = Optional.ofNullable(getById(dto.getCode())).orElseThrow(() -> new ApiException("ID不存在,不能更新"));
        final ThirdParty model = dtoToModel(dto);
        try {
            if (super.updateById(model)) {
                return modelToDto(model);
            }
            throw new ApiException("更新失败");
        } finally {
            clearCache(historyModel);
            historyModel.freeData();
            model.freeData();
        }
    }

    /**
     * 批量添加
     *
     * @param mList 数据库对象
     */
    @Transactional(rollbackFor = Exception.class)
    public void saveBatch(List<ThirdParty> mList) {
        try {
            if (!super.saveBatch(mList)) {
                throw new ApiException("批量添加失败");
            }
        } finally {
            clearCaches(mList);
        }
    }

    /**
     * 批量更新
     *
     * @param mList 数据库对象
     */
    @Transactional(rollbackFor = Exception.class)
    public void updateBatchIds(List<ThirdParty> mList) {
        final List<String> ids = mList.stream().map(ThirdParty::getCode).collect(toList());
        final List<ThirdParty> historyModelList = checkIds(ids);
        try {
            if (!super.updateBatchById(mList)) {
                throw new ApiException("批量更新失败");
            }
        } finally {
            clearCaches(historyModelList);
            historyModelList.clear();
            ids.clear();
        }
    }

    /**
     * 删除，单个
     *
     * @param code 主键
     */
    @Transactional(rollbackFor = Exception.class)
    public void removeById(String code) {
        final ThirdParty historyModel = Optional.ofNullable(getById(code)).orElseThrow(() -> new ApiException("数据不存在,不能删除"));
        try {
            if (!super.removeById(code)) {
                throw new ApiException("删除失败");
            }
        } finally {
            clearCache(historyModel);
            historyModel.freeData();
        }
    }

    /**
     * 批量删除
     *
     * @param ids 主键集合
     */
    @Transactional(rollbackFor = Exception.class)
    public void removeBatchIds(List<String> ids) {
        final List<ThirdParty> historyModelList = checkIds(ids);
        try {
            if (!super.removeByIds(ids)) {
                throw new ApiException("批量删除失败");
            }
        } finally {
            clearCaches(historyModelList);
            historyModelList.clear();
        }
    }

    /**
     * 根据传入对象构建查询参数，判断是否存在
     *
     * @param dto 参数
     * @return 存在为true
     */
    @Transactional(readOnly = true)
    public Boolean isExists(ThirdPartyDTO dto) {
        final QueryWrapper<ThirdParty> queryWrapper = createQueryWrapper(dto);
        final boolean exists = super.count(queryWrapper) > 0;
        queryWrapper.clear();
        return exists;
    }

    /**
     * 根据主键判断数据是否存在
     *
     * @param code 主键
     * @return 存在为true
     */
    @Transactional(readOnly = true)
    public Boolean isExistsById(String code) {
        boolean exists = false;
        if (StringUtils.hasLength(code)) {
            final ThirdParty model = getById(code);
            exists = (null != model);
            if (exists) {
                model.freeData();
            }
        }
        return exists;
    }

    /**
     * 根据主键获取数据
     *
     * @param code 主键
     * @return 单条记录
     */
    @Transactional(readOnly = true)
    public ThirdParty getById(String code) {
        final String key = RedisTemplateUtil.getRedisCacheKey(PREFIX_ID, code);
        return redisTemplateUtil.get(new CacheKey().setKey(key).setTimeout(ConstantDto.REDIS_TIME_OUT_DAY), () -> super.getById(code), ThirdParty.class);
    }

    /**
     * 根据主键获取数据
     *
     * @param code 主键
     * @return 单条记录，转换成页面对象
     */
    @Transactional(readOnly = true)
    public ThirdPartyDTO getByIdDto(String code) {
        return Optional.ofNullable(getById(code)).map(this::modelToDto).orElse(null);
    }

    /**
     * 根据条件查询单条数据
     *
     * @param dto 查询参数
     * @return 单条记录
     */
    @Transactional(readOnly = true)
    public ThirdParty getOne(ThirdPartyDTO dto) {
        final QueryWrapper<ThirdParty> queryWrapper = createQueryWrapper(dto);
        ThirdParty model = super.getOne(queryWrapper);
        queryWrapper.clear();
        return model;
    }

    /**
     * 根据条件查询单条数据
     *
     * @param dto 查询参数
     * @return 单条记录，转换成页面对象
     */
    @Transactional(readOnly = true)
    public ThirdPartyDTO getOneDto(ThirdPartyDTO dto) {
        return Optional.ofNullable(getOne(dto)).map(this::modelToDto).orElse(null);
    }

    /**
     * 根据条件查询列表数据
     *
     * @param dto 查询参数
     * @return 列表数据
     */
    @Transactional(readOnly = true)
    public List<ThirdParty> getList(ThirdPartyDTO dto) {
        final QueryWrapper<ThirdParty> queryWrapper = createQueryWrapper(dto);
        final List<ThirdParty> list = super.list(queryWrapper);
        queryWrapper.clear();
        return list;
    }

    /**
     * 根据条件查询列表数据
     *
     * @param dto 查询参数
     * @return 列表数据，转换成页面对象
     */
    @Transactional(readOnly = true)
    public List<ThirdPartyDTO> getListDto(ThirdPartyDTO dto) {
        return modelToDtoList(getList(dto));
    }

    /**
     * 分页数据
     *
     * @param page 分页对象
     * @param dto  查询参数
     * @return 分页信息
     */
    @Transactional(readOnly = true)
    public IPage<ThirdPartyDTO> page(Page<ThirdParty> page, ThirdPartyDTO dto) {
        final QueryWrapper<ThirdParty> queryWrapper = createQueryWrapper(dto);
        Page<ThirdParty> iPage = baseMapper.selectPage(page, queryWrapper);
        Optional.ofNullable(iPage.getRecords()).ifPresent(this::modelToDtoList);
        queryWrapper.clear();
        return iPage.convert(this::modelToDto);
    }

    /**
     * 清空缓存
     *
     * @param model 数据库对象
     */
    public void clearCache(ThirdParty model) {
        if (null == model) {
            return;
        }
        redisTemplateUtil.deleteByKey(RedisTemplateUtil.getRedisCacheKey(PREFIX_ID, model.getCode()));
    }

    /**
     * 批量清空缓存
     *
     * @param list 数据库对象
     */
    public void clearCaches(List<ThirdParty> list) {
        if (null == list || list.isEmpty()) {
            return;
        }
        List<String> ids = list.stream().map(model -> RedisTemplateUtil.getRedisCacheKey(PREFIX_ID, model.getCode())).collect(toList());
        if (!ids.isEmpty()) {
            redisTemplateUtil.deleteByKey(ids);
            ids.clear();
        }
    }

    /**
     * 数据库对象转页面对象
     *
     * @param model 数据库对象
     * @return 页面对象
     */
    public ThirdPartyDTO modelToDto(ThirdParty model) {
        if (null == model) {
            return null;
        }
        final ThirdPartyDTO dto = new ThirdPartyDTO();
        dto.setCode(model.getCode());
        dto.setName(model.getName());
        dto.setAppid(model.getAppid());
        dto.setRedirectUrl(model.getRedirectUrl());
        dto.setParams(model.getParams());
        dto.setAddTime(model.getAddTime());
        dto.setUpdateTime(model.getUpdateTime());
        dto.setAppCategory(model.getAppCategory());
        dto.setParamType(model.getParamType());
        dto.setOperator(model.getOperator());
        dto.setOrgId(model.getOrgId());
        buildDisplay(dto);
        return dto;
    }

    /**
     * 页面对象转数据库对象
     *
     * @param dto 页面对象
     * @return 数据库对象
     */
    public ThirdParty dtoToModel(ThirdPartyDTO dto) {
        if (null == dto) {
            return null;
        }
        final ThirdParty model = new ThirdParty();
        model.setCode(dto.getCode());
        model.setName(dto.getName());
        model.setAppid(dto.getAppid());
        model.setRedirectUrl(dto.getRedirectUrl());
        model.setParams(dto.getParams());
        model.setAddTime(dto.getAddTime());
        model.setUpdateTime(dto.getUpdateTime());
        model.setAppCategory(dto.getAppCategory());
        model.setParamType(dto.getParamType());
        model.setOperator(dto.getOperator());
        model.setOrgId(dto.getOrgId());
        return model;
    }

    /**
     * 码表转换
     *
     * @param dto 页面对象
     */
    public void buildDisplay(ThirdPartyDTO dto) {
        dto.setAppCategoryDisplay(MbUtil.idToDisplay(dto.getAppCategory()));
        dto.setParamTypeDisplay(MbUtil.idToDisplay(dto.getParamType()));
        dto.setOperatorDisplay(loginUserService.idToDisplay(dto.getOperator()));
        dto.setOrgIdDisplay(orgService.idToDisplay(dto.getOrgId()));
    }

    /**
     * 数据库对象转页面对象
     *
     * @param list 数据库对象
     * @return 页面对象
     */
    public List<ThirdPartyDTO> modelToDtoList(List<ThirdParty> list) {
        return Optional.ofNullable(list).orElse(Collections.emptyList()).stream().map(this::modelToDto).collect(toList());
    }

    /**
     * 页面对象转数据库对象
     *
     * @param list 页面对象
     * @return 数据库对象
     */
    public List<ThirdParty> dtoToModelList(List<ThirdPartyDTO> list) {
        return Optional.ofNullable(list).orElse(Collections.emptyList()).stream().map(this::dtoToModel).collect(toList());
    }

    /**
     * 拼装查询参数，除了主键的其他字段会自动加入进来
     *
     * @param dto 查询参数，不为空的属性会被自动加入
     */
    public QueryWrapper<ThirdParty> createQueryWrapper(ThirdPartyDTO dto) {
        final QueryWrapper<ThirdParty> queryWrapper = new QueryWrapper<>();
        if (null != dto) {
            if (StringUtils.hasLength(dto.getCode())) {
                queryWrapper.eq("code", dto.getCode());
            }
            if (StringUtils.hasLength(dto.getName())) {
                queryWrapper.like("name", dto.getName());
            }
            if (StringUtils.hasLength(dto.getAppCategory())) {
                queryWrapper.eq("app_category", dto.getAppCategory());
            }
            if (StringUtils.hasLength(dto.getParamType())) {
                queryWrapper.eq("param_type", dto.getParamType());
            }
        }
        queryWrapper.orderByDesc("add_time");
        return queryWrapper;
    }

    /**
     * 批量检查主键在数据库中是否存在
     *
     * @param ids 主键集合
     * @return 主键查询出来的集合
     */
    public List<ThirdParty> checkIds(List<String> ids) {
        if (null == ids || ids.isEmpty()) {
            throw new ApiException("ids不能为空");
        }
        final List<ThirdParty> list = super.listByIds(ids);
        if (list.isEmpty()) {
            throw new ApiException("ids不存在" + Arrays.toString(ids.toArray()));
        }
        if (ids.size() == list.size()) {
            return list;
        }
        List<String> dbIds = list.stream().map(ThirdParty::getCode).collect(toList());
        final List<String> notExistsList = ids.stream().filter(id -> !dbIds.contains(id)).collect(toList());
        dbIds.clear();
        if (!notExistsList.isEmpty()) {
            throw new ApiException("ids不存在" + Arrays.toString(notExistsList.toArray()));
        }
        return list;
    }

    private void validator(ThirdPartyDTO dto) {
        if (MbUtil.isNotExists(dto.getAppCategory(), BaseTableEnum.APP_CATEGORY)) {
            throw new ApiException("应用分类值错误");
        }
        if (StringUtils.hasLength(dto.getParamType())) {
            if (MbUtil.isNotExists(dto.getParamType(), BaseTableEnum.PARAM_TYPE)) {
                throw new ApiException("参数类型值错误");
            }
        }
    }

    @Transactional(readOnly = true)
    public String idToDisplay(String thirdParty) {
        if (StringUtils.hasLength(thirdParty)) {
            return Optional.ofNullable(getById(thirdParty)).map(ThirdParty::getName).orElse("");
        }
        return "";
    }
}